package com.common.app.data.parser

import com.common.app.data.bean.Response
import com.common.base.data.ListCommon
import rxhttp.wrapper.annotation.Parser
import rxhttp.wrapper.entity.ParameterizedTypeImpl
import rxhttp.wrapper.exception.ParseException
import rxhttp.wrapper.parse.TypeParser
import rxhttp.wrapper.utils.convert
import rxhttp.wrapper.utils.convertTo
import java.io.IOException
import java.lang.reflect.Type

/**
 * 输入T,输出T,并对code统一判断
 *
 * 如果使用协程发送请求，wrappers属性可不设置，设置了也无效
 */
@Parser(name = "Response", wrappers = [ListCommon::class])
open class ResponseParser<T> : TypeParser<T> {
    /**
     * 此构造方法适用于任意Class对象，但更多用于带泛型的Class对象，如：List<Student>
     *
     * 用法:
     * Java: .asParser(new ResponseParser<List<Student>>(){})
     * Kotlin: .asParser(object : ResponseParser<List<Student>>() {})
     *
     * 注：此构造方法一定要用protected关键字修饰，否则调用此构造方法将拿不到泛型类型
     */
    protected constructor() : super()

    /**
     * 此构造方法仅适用于不带泛型的Class对象，如: Student.class
     *
     * 用法
     * Java: .asParser(new ResponseParser<>(Student.class))   或者  .asResponse(Student.class)
     * Kotlin: .asParser(ResponseParser(Student::class.java)) 或者  .asResponse<Student>()
     */
    constructor(type: Type) : super(type)

    @Throws(IOException::class)
    override fun onParse(response: okhttp3.Response): T {
//        val type: Type = ParameterizedTypeImpl[Response::class.java, types] //获取泛型类型
        val type: Type = ParameterizedTypeImpl.getParameterized(Response::class.java, *types) //获取泛型类型
        val data: Response<T> = response.convert(type)
        var t: T? = data.data //获取data字段

//        if (data.getCode() == 300) {       // 账号退出登录
//            throw new AccountExitExcep(data.getCode(), data.getMsg());
//        } else if (data.getCode() == 350) {       // 账号被封禁
//            throw new AccountBanExcep(data.getCode(), data.getMsg());
//        } else if (data.getCode() == 380) {       // 地区异常
//            throw new LocalErrorExcep(data.getCode(), data.getMsg());
//        } else if (data.getCode() != 200 || t == null) {//这里假设code不等于200，代表数据不正确，抛出异常
        //        if (data.getCode() == 300) {       // 账号退出登录
//            throw new AccountExitExcep(data.getCode(), data.getMsg());
//        } else if (data.getCode() == 350) {       // 账号被封禁
//            throw new AccountBanExcep(data.getCode(), data.getMsg());
//        } else if (data.getCode() == 380) {       // 地区异常
//            throw new LocalErrorExcep(data.getCode(), data.getMsg());
//        } else if (data.getCode() != 200 || t == null) {//这里假设code不等于200，代表数据不正确，抛出异常
        if (t == null && types == String::class.java) {
            t = "" as T
        }
        if (data.code != 200) { //这里假设code不等于200，代表数据不正确，抛出异常
            throw ParseException(data.code.toString(), data.msg, response)
        }
        return t!!


//        val data: Response<T> = response.convertTo(Response::class, *types)
//        var t = data.data //获取data字段
//        if (t == null && types[0] === String::class.java) {
//            /*
//             * 考虑到有些时候服务端会返回：{"errorCode":0,"errorMsg":"关注成功"}  类似没有data的数据
//             * 此时code正确，但是data字段为空，直接返回data的话，会报空指针错误，
//             * 所以，判断泛型为String类型时，重新赋值，并确保赋值不为null
//             */
//            @Suppress("UNCHECKED_CAST")
//            t = data.msg as T
//        }
//        if (data.code != 0 || t == null) { //code不等于0，说明数据不正确，抛出异常
//            throw ParseException(data.code.toString(), data.msg, response)
//        }
//        return t
    }
}